package com.atguigu.java4;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

/**
 * @author yihu
 * @create 2021-06-17 15:33
 * @email:1057202547@qq.com
 * @dec: 动态代理
 */

interface  Human{
    String getBelief();
    void eat(String food);
}

//被代理类
class SuperMan implements  Human {
    @Override
    public String getBelief() {
        return "I belief I can fly!";
    }
    @Override
    public void eat(String food) {
        System.out.println("我喜欢吃" + food);
    }
}

/*
   要想实现动态代理，需要解决的问题？
   问题一：如何根据加载到内存中的被代理类，动态的创建一个代理类及其对象。
   问题二：当通过代理类的对象调用方法a时，如何动态的去调用被代理类中的同名方法a。
 */
 //代理类的工厂
 class  ProxyFactory {
    /**
     * 调用此方法，返回一个代理类的对象。解决问题一
     * @param obj 被代理类的对象
     * @return 代理类的对象
     */
     public  static  Object getProxyInstance(Object obj){
         MyInvocationHandler handler = new MyInvocationHandler();
         handler.bind(obj);
         //创建一个代理类
         return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),handler);
     }
 }

 //创建一个InvocationHandler 接口的类

 class  MyInvocationHandler implements InvocationHandler{

     private  Object obj; //被代理类的对象进行赋值

     //被代理类进行赋值
     public  void bind(Object obj){
         this.obj = obj;
     }

     //当我们通过代理类的对象，调用方法a时，就会自动的调用如下的方法：invoke()
     //将被代理类要执行的方法a的功能就声明在invoke()中
     @Override
     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
         //method:即为代理类对象调用的方法，此方法也就作为了被代理类对象要调用的方法
         //obj:被代理类的对象
         Object returnValue = method.invoke(obj, args);
         //上述方法的返回值就作为当前类中的invoke()的返回值。
         return returnValue;
     }
 }

public class ProxyTest {
    public static void main(String[] args) {
        //返回的代理对象是一个实现了Human 接口的类 的代理类proxyInstance
        //new SuperMan() 被代理类
        Human proxyInstance =(Human) ProxyFactory.getProxyInstance(new SuperMan());

        //当通过代理类对象调用方法时，会自动的调用被代理类中同名的方法
        String belief = proxyInstance.getBelief();
        System.out.println(belief);
        proxyInstance.eat("四川麻辣烫");


        NikeClothFactory nikeClothFactory = new NikeClothFactory();
        ClothFactory  proxyInstance2 =(ClothFactory) ProxyFactory.getProxyInstance(nikeClothFactory);
        proxyInstance2.produceCloth();

    }
}